1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
#![allow(non_snake_case)]

use crate::co;
use crate::oleaut::ffi;
use crate::prelude::*;

/// [`PROPVARIANT`](https://learn.microsoft.com/en-us/windows/win32/api/propidlbase/ns-propidlbase-propvariant)
/// struct.
///
/// Automatically calls
/// [`PropVariantClear`](https://learn.microsoft.com/en-us/windows/win32/api/combaseapi/nf-combaseapi-propvariantclear)
/// when the object goes out of scope.
///
/// The [`Default`](std::default::Default) implementation returns a
/// [`co::VT::EMPTY`](crate::co::VT::EMPTY) value.
#[repr(C)]
pub struct PROPVARIANT {
	vt: co::VT,
	wReserved1: u16,
	wReserved2: u16,
	wReserved3: u16,
	data: [u8; 16],
}

impl Drop for PROPVARIANT {
	fn drop(&mut self) {
		if self.vt() != co::VT::EMPTY {
			unsafe { ffi::PropVariantClear(self as *mut _ as _); } // ignore errors
		}
	}
}

impl Default for PROPVARIANT {
	fn default() -> Self {
		unsafe { std::mem::zeroed::<Self>() } // PropVariantInit() is just a macro
	}
}

impl oleaut_Variant for PROPVARIANT {
	fn raw(&self) -> &[u8; 16] {
		&self.data
	}

	unsafe fn from_raw(vt: co::VT, data: &[u8]) -> Self {
		let mut obj = Self::default();
		obj.vt = vt;
		data.iter()
			.zip(&mut obj.data)
			.for_each(|(src, dest)| *dest = *src);
		obj
	}

	fn vt(&self) -> co::VT {
		self.vt
	}
}

impl PROPVARIANT {
	/// Creates a new object holding an `i64` value.
	#[must_use]
	pub fn new_i64(val: i64) -> Self {
		unsafe { Self::from_raw(co::VT::I8, &val.to_ne_bytes()) }
	}

	/// If the object holds an `i64` value, returns it, otherwise `None`.
	#[must_use]
	pub fn i64(&self) -> Option<i64> {
		if self.vt() == co::VT::I8 {
			Some(i64::from_ne_bytes(self.raw()[..8].try_into().unwrap()))
		} else {
			None
		}
	}

	/// Creates a new object holding an `u64` value.
	#[must_use]
	pub fn new_u64(val: u64) -> Self {
		unsafe { Self::from_raw(co::VT::UI8, &val.to_ne_bytes()) }
	}

	/// If the object holds an `u64` value, returns it, otherwise `None`.
	#[must_use]
	pub fn u64(&self) -> Option<u64> {
		if self.vt() == co::VT::UI8 {
			Some(u64::from_ne_bytes(self.raw()[..8].try_into().unwrap()))
		} else {
			None
		}
	}
}